-
Notifications
You must be signed in to change notification settings - Fork 279
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for L1 MPT #1104
base: develop
Are you sure you want to change the base?
Add support for L1 MPT #1104
Conversation
WalkthroughThe pull request introduces enhancements to the Ethereum client's state management, focusing on the integration of the Scroll network and stateless execution capabilities. Key changes include the addition of a new flag for configuring Merkle Patricia Trie (MPT) options, modifications to state validation logic, and the introduction of methods for managing disk state roots. The API is expanded to support execution witnesses, and new witness tracking mechanisms are implemented. These changes aim to improve state validation, enhance blockchain configuration flexibility, and provide insights into state transitions. Changes
Possibly related PRs
Suggested reviewers
Poem
✨ Finishing Touches
Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments. CodeRabbit Commands (Invoked using PR comments)
Other keywords and placeholders
CodeRabbit Configuration File (
|
9716dfa
to
bc9dc5d
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 1
🧹 Nitpick comments (27)
core/stateless/witness.go (2)
49-69
: Check parent header retrieval logicWhen
chain
is not nil, the code retrieves the parent header (parent := chain.GetHeader(...)
). Ifparent
is nil for any reason (database inconsistency or missing data), an error is returned. Consider logging the block hash or other context for easier debugging, especially in long-running systems.
83-88
: Check memory usage for code collectionStoring code in a map keyed by the string representation (
w.Codes[string(code)]
) is simple, but storing large byte slices in string form can result in heightened memory usage or possible collisions. If large codes are anticipated, you might want to store hashes or references instead.core/state/database.go (2)
108-110
: Document potential performance impact of returning large setsThe new interface method
Witness() map[string]struct{}
may produce large sets if the trie is big. Document usage expectations or provide a streaming alternative if performance or memory is a concern.
142-144
: Log or handle errors encountered while reading disk rootCurrently, if
rawdb.ReadDiskStateRoot(...)
returns an error, it is silently ignored and the existingroot
remains. Adding debug logs or metrics in theerr != nil
case can help troubleshoot issues (e.g., partial corruption or I/O problems).trie/zk_trie_test.go (1)
289-338
: Global state usage in tests may cause concurrency pitfallsGlobal vars (e.g.,
accountsLeft
) can lead to race conditions if tests run in parallel. Wrap them inside the test struct or use test-level counters to avoid altering shared state, ensuring thread-safe test environments.trie/trie.go (3)
65-67
: Document or enforce concurrency restrictions ontracer
The
tracer
field withinTrie
indicates added instrumentation for tracking node access. If the trie is used concurrently, clarify thatt.tracer
is thread-safe or enforce single-thread usage of the trie to prevent race conditions.
320-324
: Manage tracer calls during node insertions/deletionsCalls to
t.tracer.onInsert(...)
andt.tracer.onDelete(...)
are scattered across the insertion and deletion paths. Ensure consistent handling of partial updates (e.g., if an insert fails midway). Consider a rollback or snapshot approach for robust error handling.Also applies to: 339-343, 396-400, 413-416, 478-482
621-631
: Assess memory footprint of witness generation
func (t *Trie) Witness() map[string]struct{}
accumulates accessed trie nodes int.tracer.accessList
. For large tries, this might be significant. If memory usage becomes problematic, consider a streaming approach or partial flushing of tracked nodes.eth/api.go (2)
326-337
: Consider returning nil witness when errors occur.Currently, if
generateWitness
returns a partially-built witness along with an error, the subsequent return passes the non-nil witness toToExecutionWitness
. Depending on the calling context, partial data might be confusing or misleading if there's an error. Consider returningnil
in case of errors to avoid partial witness representations.
365-394
: Add more context about potential partial failures.The function
testWitness
properly checks if thediskRoot
exists and reassignsstateRoot
orpostStateRoot
. The error strings are general, but you might consider including additional context (e.g., mismatch details or expected vs. computed root) to help troubleshoot witness failures.core/state/statedb.go (3)
300-302
: Avoid double retrieval of code inGetCode
.The code snippet calls
stateObject.Code(s.db)
both when adding to the witness and again when returning. For large contracts, this might induce overhead. A small refactor to store it in a local variable is recommended:code := stateObject.Code(s.db) if s.witness != nil { - s.witness.AddCode(stateObject.Code(s.db)) + s.witness.AddCode(code) } return code
311-314
: Consider caching code inGetCodeSize
.Although the function fetches only the code size, calling
stateObject.Code(s.db)
inAddCode
can be expensive. If the logic truly needs the code bytes, caching it or confirming necessity may be worthwhile.
933-939
: Repeated addition of storage trie witnesses.We see the
s.witness.AddState(obj.trie.Witness())
call repeated before and afterobj.updateRoot(...)
. Possibly unify or clarify if both calls are indeed necessary, or combine them if feasible.params/config.go (1)
899-902
: Doc mismatch inIsEuclid
.The comment says “returns whether num is … the Darwin fork block,” but the implementation references
EuclidTime
. Update doc comment to match the Euclid fork context to avoid confusion.core/stateless/database.go (2)
47-55
: Improve readability with direct iteration over codes map values.
Instead of iterating overmap[string]struct{}
keys and converting them to[]byte
, it might be more expressive to store the code as[]byte
in the map (assuming the data is originally in[]byte
format). This eliminates the need to convert fromstring
to[]byte
.
57-65
: Avoid potential collisions in hashing.
While Keccak-256 collisions are extremely unlikely, ensure no mismatch occurs if a future enhancement tries to unify code and state node writes. Using distinct storage keys for codes vs. nodes (like a prefix) is prudent.trie/tracer.go (2)
45-48
: Consider more descriptive field names or doc comments.
Whileinserts
,deletes
, andaccessList
are straightforward, additional docstrings clarifying their role might help maintainers.
88-93
: Align tracer lifecycle with other components.
After callingreset
, ensure other references tot.inserts
ort.deletes
won't inadvertently reintroduce stale data.cmd/geth/usage.go (1)
53-53
: Add usage information forScrollMPTFlag
.The new flag is introduced without explicit usage text. Consider adding a brief description of its effect on the MPT state mechanism to guide users.
trie/zk_trie.go (3)
237-243
: Validate error handling when retrieving the ZkTrie root.
panic("CountLeaves cannot get root")
may cause a crash in production if the root retrieval fails. Consider returning an error instead to fail gracefully in edge cases (e.g., partial database corruption).
245-260
: Avoid panics for recoverable I/O errors.
panic("countLeaves cannot get rootNode")
may terminate the node. Consider translating I/O errors into typed errors or logging them for safer recoverability.
262-265
: Implement theWitness
method or provide a TODO comment.Leaving this method as a
panic("not implemented")
can hamper the broader witness functionality. If you plan to add it soon, consider adding a TODO clarifying the intended approach.cmd/geth/main.go (1)
153-153
: Add usage examples or clarifications.
utils.ScrollMPTFlag
is added here. Consider documenting the effect of enabling it in help messages or logs to guide users, e.g., “Enables L1 MPT features.”core/state/state_object.go (2)
503-514
: Error handling improvement inCodeSize
.
- The approach gracefully handles the scenario where
KeccakCodeHash
is empty.- However, storing the error in
dbErr
can mask subsequent errors, since only the first is recorded. Consider logging or ensuring it doesn’t impede diagnosing multiple retrieval issues.
547-547
: Document the panic reason.Replacing the previous implementation with
panic("shouldn't be called")
is valid if truly unreachable. Provide a descriptive explanation or reference in code comments to clarify why it's never expected to be invoked.miner/scroll_worker.go (1)
812-815
: Validate pre-fork and post-fork states.The condition
canCommitState := w.chainConfig.Scroll.UseZktrie != w.chainConfig.IsEuclid(w.current.header.Time)
ensures the block commit is skipped unless Zktrie usage aligns with the Euclid fork timing. Verify that off-by-one scenarios and minor clock discrepancies do not cause indefinite skipping, potentially blocking the chain.cmd/utils/flags.go (1)
1905-1912
: Consider refactoring duplicate configuration logic.The Zktrie configuration logic is identical for both Scroll Sepolia and mainnet networks. Consider extracting this into a helper function to reduce code duplication and improve maintainability.
+func configureZktrie(ctx *cli.Context, cfg *ethconfig.Config) { + cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) + if cfg.Genesis.Config.Scroll.UseZktrie { + // disable pruning + if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { + log.Crit("Must use --gcmode=archive") + } + log.Info("Pruning disabled") + cfg.NoPruning = true + } +}Then use it in both places:
case ctx.GlobalBool(ScrollSepoliaFlag.Name): // ... existing code ... - cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) - if cfg.Genesis.Config.Scroll.UseZktrie { - // disable pruning - if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { - log.Crit("Must use --gcmode=archive") - } - log.Info("Pruning disabled") - cfg.NoPruning = true - } + configureZktrie(ctx, cfg) case ctx.GlobalBool(ScrollFlag.Name): // ... existing code ... - cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) - if cfg.Genesis.Config.Scroll.UseZktrie { - // disable pruning - if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { - log.Crit("Must use --gcmode=archive") - } - log.Info("Pruning disabled") - cfg.NoPruning = true - } + configureZktrie(ctx, cfg)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (27)
cmd/geth/main.go
(1 hunks)cmd/geth/usage.go
(1 hunks)cmd/utils/flags.go
(3 hunks)core/block_validator.go
(1 hunks)core/blockchain.go
(3 hunks)core/genesis.go
(1 hunks)core/rawdb/accessors_state.go
(1 hunks)core/rawdb/schema.go
(2 hunks)core/state/database.go
(2 hunks)core/state/state_object.go
(2 hunks)core/state/statedb.go
(8 hunks)core/stateless/database.go
(1 hunks)core/stateless/encoding.go
(1 hunks)core/stateless/witness.go
(1 hunks)core/types/state_account.go
(1 hunks)eth/api.go
(3 hunks)internal/web3ext/web3ext.go
(2 hunks)light/trie.go
(1 hunks)miner/scroll_worker.go
(1 hunks)params/config.go
(8 hunks)rollup/pipeline/pipeline.go
(1 hunks)trie/database.go
(2 hunks)trie/secure_trie.go
(2 hunks)trie/tracer.go
(1 hunks)trie/trie.go
(9 hunks)trie/zk_trie.go
(1 hunks)trie/zk_trie_test.go
(2 hunks)
🔇 Additional comments (47)
core/stateless/witness.go (4)
75-80
: Validate retrieved headers in the loop
This loop appends headers by repeatedly calling w.chain.GetHeader(...)
without checking for a nil return. If the chain database is missing headers for any reason, this code will panic on the next iteration. Suggest verifying that each retrieved header is non-nil and returning an error or handling the missing header appropriately.
90-99
: Concurrent insertion is well-protected
The function uses a mutex (w.lock
) to protect against concurrent writes to w.State
, which is good. The approach of using maps.Copy
is straightforward and correct, ensuring no partial updates occur if multiple goroutines call AddState
.
101-114
: Deep copy approach is correct and coherent
Cloning slices and maps through slices.Clone
and maps.Clone
is a clean approach to guaranteeing an independent copy of the witness state. This is especially beneficial if you need an immutable snapshot of the witness at any given state.
120-122
: Ensure non-empty Headers slice before indexing
return w.Headers[0].Root
indexes the first header in the slice. If w.Headers
happens to be empty (for example, if chain
was unavailable), this will panic. Consider adding a precondition check (e.g., if len(w.Headers) == 0 { ... }
) to handle unexpected states more gracefully.
trie/zk_trie_test.go (2)
29-38
: Import changes look fine
The additional require
import from github.com/stretchr/testify/require
and other minor import changes are standard for improved test assertions and do not introduce any apparent issues.
272-287
: Avoid embedding absolute paths in test code
The test function TestEquivalence
references specific local paths such as /Users/omer/Documents/go-ethereum/l2geth-datadir/geth/chaindata
. This limits portability and may break in CI environments. Consider parameterizing these paths or using temp directories.
rollup/pipeline/pipeline.go (1)
231-231
: Prefetcher invocation looks consistent
Invoking p.state.StartPrefetcher("miner", nil)
with the extra parameter is consistent with the approach in other parts of the code. This matches the method signature changes observed in related files.
trie/trie.go (1)
618-619
: Tracer reset for subsequent usage
t.tracer.reset()
is invoked in t.Reset()
. This is fine if the intention is to clear all previous tracking data. Just confirm that no external references rely on the old tracer state once the reset is done.
eth/api.go (7)
37-39
: No issues with the new imports.
The import of "github.com/scroll-tech/go-ethereum/core/stateless"
appears straightforward and does not introduce any obvious conflicts or redundancies.
339-363
: Implementation of generateWitness
looks good.
The error handling at each step (creating the witness, retrieving the parent state, processing, and validating) is consistent. Returning (witness, testWitness(...))
ensures immediate cross-validation.
396-403
: New ExecutionWitness
type is straightforward.
This struct properly exposes headers, codes, and state in an easy-to-serialize format. No issues spotted.
405-413
: transformMap
is cleanly implemented.
Using the keccak hash of the item as the map key and storing the hex-encoded node as the value is a clear approach.
415-424
: ToExecutionWitness
accurately transforms the internal witness representation.
The function provides a concise mapping of w.Codes
and w.State
into the final JSON representation. This is efficient for cross-client compatibility.
965-968
: DiskAndHeaderRoot
struct.
Straightforward struct addition, properly typed fields.
970-990
: DiskRoot
method design is straightforward.
The method checks for a stored disk root and falls back to the header root if none is found. Consider logging or returning the encountered error from rawdb.ReadDiskStateRoot
if needed for debugging or further handling.
core/state/statedb.go (5)
32-32
: New import of github.com/scroll-tech/go-ethereum/core/stateless
.
No immediate concerns. This aligns with the new witness functionality.
110-112
: Confirm concurrency safety for the new witness
field.
If multiple goroutines may call StartPrefetcher
or manipulate witness
, ensure synchronization or clarify single-threaded usage to avoid potential data races.
166-174
: StartPrefetcher
changes approved.
Warmly closing any previously running prefetcher and then setting s.witness
is clear. The logic for prefetcher = newTriePrefetcher(...)
remains consistent.
950-959
: Approved additional witness logic.
Gathering the storage trie for objects that have originStorage
is consistent with the approach. Implementation looks solid.
991-997
: Final witness addition in IntermediateRoot
.
Appending the account trie witness here is appropriately placed. The code is straightforward and ensures the final root’s witness is included.
params/config.go (6)
32-41
: Genesis hash and state additions look correct.
These newly defined constants for the Scroll networks do not raise any conflict.
345-345
: GenesisStateRoot
assignment for Scroll Sepolia.
No issues. The reference to ScrollSepoliaGenesisState
is consistent with the new config fields.
386-386
: GenesisStateRoot
assignment for Scroll Mainnet.
Same approach as Sepolia, logically consistent.
640-640
: New EuclidTime
field.
Adding the Euclid time-based fork is in line with existing time-based forks. No immediate concerns.
669-671
: GenesisStateRoot *common.Hash
in ScrollConfig
.
Well-defined pointer for capturing the MPT-based genesis state root. Suits the new requirements.
1116-1116
: IsEuclid
integration into Rules
.
Adding the boolean is consistent with the new time-based fork logic.
core/types/state_account.go (1)
34-35
: RLP Ignoring PoseidonCodeHash
and CodeSize
.
Marking these fields with rlp:"-"
prevents them from being included in the main account encoding. This is correct if they’re externally derived or not part of the consensus object.
core/stateless/encoding.go (1)
1-76
: New file for witness encoding/decoding.
- License & Package Declarations (lines 1-18): No issues. The license header and package declaration are standard.
- Imports (lines 19-24): Straightforward; includes
rlp
andtypes
. toExtWitness
/fromExtWitness
(lines 26-40, 42-55): Good separation of the external vs. internal representation. The approach of converting maps to slices and vice versa is clean.- RLP Encode/Decode (lines 57-61, 62-69): Implementation is standard. RLP into an
extWitness
intermediate, then converted to/from the internalWitness
. This fosters a simpler code path for serialization. extWitness
Definition (lines 71-76): Straightforward structure for RLP transport.
Overall, the new file cleanly implements witness serialization with no evident performance or correctness issues.
core/stateless/database.go (2)
19-24
: Consider grouping imports logically, or adding minimal usage comments.
Imports are quite straightforward here. Just ensure each imported package is used consistently throughout the file and keep them minimal if possible.
36-67
: Ensure consistent error checking when writing to the ephemeral DB.
Currently, the code writes headers, codes, and node data without checking for potential errors from the rawdb
layer. If any write fails, an error would remain silent. Consider either:
- Handling the returned errors from
WriteHeader
,WriteCode
, andWriteTrieNode
, or - Logging them if complete error handling is not required.
core/rawdb/accessors_state.go (1)
98-110
: Handle partial database writes more gracefully.
WriteDiskStateRoot and ReadDiskStateRoot rely on db.Put/db.Get but do not consider partial writes or concurrent usage. If concurrency is expected, ensure your underlying DB can handle it, or protect these calls with appropriate synchronization.
trie/tracer.go (3)
66-75
: Handle reinserted and resurrected nodes with caution.
The logic for removing an item from deletes
if it is resurrected seems correct. Just ensure no unexpected concurrency issues occur if multiple insert/delete calls overlap.
95-106
: Deep copy overhead and memory usage.
When copy()
is called, a full deep copy is performed, which could be expensive for large tries. Verify that copy()
won't degrade performance in large-scale usage or that usage patterns remain minimal.
108-122
: Confirm filtering in deletedNodes
if embedded nodes never appear in accessList
.
Some embedded node deletions might not be recognized as “formerly accessible.” Confirm the behavior is correct when partial expansions occur and not all nodes have been read.
trie/secure_trie.go (2)
193-193
: Ensure deep copying of the tracer.
Copying the tracer is beneficial for preserving the original SecureTrie
's state. Verify that t.trie.tracer.copy()
performs a deep copy to avoid data races if shared among multiple SecureTrie
instances.
226-229
: Document concurrency implications for the new Witness
method.
Returning a map of accessed nodes is useful for debugging or proof generation. However, consider clarifying concurrency recommendations in the method’s docstring to ensure thread-safety or potential usage constraints.
core/block_validator.go (1)
229-230
: Verify correctness of the state root validation toggle.
shouldValidateStateRoot := v.config.Scroll.UseZktrie != v.config.IsEuclid(header.Time)
introduces a conditional skip of state root checks. Confirm that skipping validation is intentional, as it may allow inconsistent states when certain flags or time conditions are met.
core/rawdb/schema.go (2)
130-130
: Consider prefix collisions.
While diskStateRootPrefix
is short and descriptive, ensure no existing prefix collisions occur. Double-check that keys using this prefix won’t inadvertently overlap with others.
318-320
: Function usage verification.
diskStateRootKey
cleanly appends diskStateRootPrefix
and headerRoot
. Verify that all call sites (for example, WriteDiskStateRoot, ReadDiskStateRoot) handle errors or empty roots consistently.
internal/web3ext/web3ext.go (2)
485-491
: Validate input for executionWitness
.
Adding the new debug_executionWitness
method is helpful. Consider clarifying input usage or adding front-end checks, since inputFormatter: [null]
does not enforce specific validations.
952-958
: Check block number range in diskRoot
.
diskRoot
uses inputDefaultBlockNumberFormatter
, which presumably handles typical edge cases. Validate that negative or extremely large values are handled gracefully downstream.
trie/database.go (1)
670-672
: Validate behavior on read error.
Here, if rawdb.ReadDiskStateRoot
returns an error, the logic silently falls back to the original node
without any logs. This may be intentional, but consider adding debug logs upon errors to diagnose potential mismatches between the in-memory and disk state roots.
core/blockchain.go (2)
1683-1683
: Check for consistent usage of the additional parameter.
Calling statedb.StartPrefetcher("chain", nil)
clarifies that no witness data is supplied. Ensure all code references expecting a non-nil pointer handle this case gracefully to avoid runtime errors or unexpected prefetch behavior.
[approve]
1820-1820
: Confirm witness parameter usage.
Similarly, statedb.StartPrefetcher("l1sync", nil)
is consistent with the prior pattern. Verify that no code path assumes a valid, non-nil witness, preventing possible nil pointer dereferences if the parameter is used in the future.
core/genesis.go (1)
325-328
: Ensure consistency of configured genesis state root.
If GenesisStateRoot
is not nil but does not match the computed root
, there may be confusion about which root is canonical. Double-check that advanced scenarios involving custom or pre-computed state roots handle discrepancies correctly, and consider logging or validating that the stored root matches the actual state root.
cmd/utils/flags.go (2)
186-189
: LGTM: Clear and well-defined flag.
The new flag follows the existing pattern and clearly describes its purpose.
1886-1893
: LGTM: Proper configuration for Scroll Sepolia network.
The code correctly:
- Sets UseZktrie based on the inverse of ScrollMPTFlag
- Enforces archive mode when Zktrie is used
- Provides clear error messages and logging
bc9dc5d
to
bef84a4
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 7
🧹 Nitpick comments (12)
core/state/state_test.go (2)
158-159
: Use consistent naming for clarity
The variablesstateDb
andstate
may lead to confusion when used interchangeably with other references to “database” vs. “state” in the codebase. Consider renaming them for clarity, for instance,db
andstateDB
.
233-234
: Consider preemptive code caching
CallingCodeSize(db)
repeatedly could introduce overhead if done in bigger loops. For tests, it’s likely negligible, but in production usage, you might consider caching or storing code sizes once fetched.core/stateless/database.go (1)
36-67
: Add error handling or guard rails for empty or nil witness
MakeHashDB()
proceeds assumingw.Headers
,w.Codes
, andw.State
are valid and non-nil. Ifw
is partially initialized or empty, the code might silently succeed or produce unexpected results. Consider adding safeguards or returning an error if the witness is incomplete.core/rawdb/accessors_state.go (1)
98-110
: Check for potential collisions or overwrites
When writing disk state roots using the parent header’s hash as the key, ensure that no collisions with other processes or future expansions in the DB scheme can occur. Also, consider handling partial writes or error states more gracefully instead of callinglog.Crit
.core/stateless/witness.go (1)
17-47
: Document concurrency rules
TheWitness
struct includes async.Mutex
for concurrency. Document which methods are safe to call concurrently (e.g., “All public methods are concurrency-safe after initialization.”) to avoid confusion for other maintainers.core/state/database.go (1)
142-144
: Consider explicit error handlingWhile the error handling works, consider being more explicit about the expected error types and logging relevant information for debugging purposes.
-if diskRoot, err := rawdb.ReadDiskStateRoot(db.db.DiskDB(), root); err == nil { +diskRoot, err := rawdb.ReadDiskStateRoot(db.db.DiskDB(), root) +if err != nil { + log.Debug("No disk state root found", "root", root, "err", err) +} else { root = diskRoot }trie/zk_trie_test.go (2)
296-316
: Refactor test helper for better maintainabilityThe
checkTrieEquality
function has multiple responsibilities and could be split into smaller, more focused functions.Consider breaking it down:
- A function to initialize tries
- A function to compare tries
- A function to track progress
This would improve readability and make the code easier to maintain.
318-330
: Add error cases to account equality checksThe account equality check only verifies the happy path. Consider adding test cases for:
- Malformed account data
- Missing accounts
- Different account types
eth/api.go (2)
339-394
: Consider adding memory management for large blocks.The witness generation and testing process could consume significant memory for blocks with many state changes. Consider adding a warning log when processing large blocks.
func generateWitness(blockchain *core.BlockChain, block *types.Block) (*stateless.Witness, error) { + if block.Transactions().Len() > 1000 { + log.Warn("Generating witness for large block", "number", block.Number(), "txs", block.Transactions().Len()) + } witness, err := stateless.NewWitness(block.Header(), blockchain)
396-424
: Consider adding version field to ExecutionWitness struct.Adding a version field would help with future compatibility and migrations.
type ExecutionWitness struct { + Version uint64 `json:"version"` Headers []*types.Header `json:"headers"` Codes map[string]string `json:"codes"` State map[string]string `json:"state"` }
core/state/statedb.go (1)
166-174
: Consider adding witness cleanup in StopPrefetcher.The witness field should be cleared when stopping the prefetcher to prevent memory leaks.
func (s *StateDB) StopPrefetcher() { if s.prefetcher != nil { s.prefetcher.close() s.prefetcher = nil } + s.witness = nil }
cmd/utils/flags.go (1)
1905-1912
: Consider refactoring the duplicated Zktrie configuration logic.The Zktrie configuration logic is duplicated between Scroll Sepolia and Scroll mainnet. Consider extracting this into a helper function to improve maintainability.
+func configureZktrie(ctx *cli.Context, cfg *ethconfig.Config) { + cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) + if cfg.Genesis.Config.Scroll.UseZktrie { + // disable pruning + if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { + log.Crit("Must use --gcmode=archive") + } + log.Info("Pruning disabled") + cfg.NoPruning = true + } +}Then use it in both network configurations:
case ctx.GlobalBool(ScrollSepoliaFlag.Name): // ... existing configuration ... - cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) - if cfg.Genesis.Config.Scroll.UseZktrie { - // disable pruning - if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { - log.Crit("Must use --gcmode=archive") - } - log.Info("Pruning disabled") - cfg.NoPruning = true - } + configureZktrie(ctx, cfg) case ctx.GlobalBool(ScrollFlag.Name): // ... existing configuration ... - cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) - if cfg.Genesis.Config.Scroll.UseZktrie { - // disable pruning - if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { - log.Crit("Must use --gcmode=archive") - } - log.Info("Pruning disabled") - cfg.NoPruning = true - } + configureZktrie(ctx, cfg)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (28)
cmd/geth/main.go
(1 hunks)cmd/geth/usage.go
(1 hunks)cmd/utils/flags.go
(3 hunks)core/block_validator.go
(1 hunks)core/blockchain.go
(3 hunks)core/genesis.go
(1 hunks)core/rawdb/accessors_state.go
(1 hunks)core/rawdb/schema.go
(2 hunks)core/state/database.go
(2 hunks)core/state/state_object.go
(2 hunks)core/state/state_test.go
(4 hunks)core/state/statedb.go
(8 hunks)core/stateless/database.go
(1 hunks)core/stateless/encoding.go
(1 hunks)core/stateless/witness.go
(1 hunks)core/types/state_account.go
(1 hunks)eth/api.go
(3 hunks)internal/web3ext/web3ext.go
(2 hunks)light/trie.go
(1 hunks)miner/scroll_worker.go
(1 hunks)params/config.go
(8 hunks)rollup/pipeline/pipeline.go
(1 hunks)trie/database.go
(2 hunks)trie/secure_trie.go
(2 hunks)trie/tracer.go
(1 hunks)trie/trie.go
(9 hunks)trie/zk_trie.go
(1 hunks)trie/zk_trie_test.go
(2 hunks)
🚧 Files skipped from review as they are similar to previous changes (14)
- cmd/geth/usage.go
- cmd/geth/main.go
- core/types/state_account.go
- miner/scroll_worker.go
- core/rawdb/schema.go
- core/block_validator.go
- light/trie.go
- trie/zk_trie.go
- core/genesis.go
- trie/database.go
- trie/trie.go
- trie/tracer.go
- core/stateless/encoding.go
- core/blockchain.go
🔇 Additional comments (19)
core/state/state_test.go (2)
205-205
: Validate return errors when calling helper functions
Ensure that compareStateObjects(...)
provides meaningful diagnostics on error scenarios. If it must fail the test, consider returning an error or capturing additional context about the failing data.
214-214
: Check for concurrency
compareStateObjects
might be shared among parallel tests. Ensure no concurrency hazards exist (e.g., shared state or external references) that might cause inconsistent test behavior.
core/stateless/witness.go (2)
49-69
: Validate parent header retrieval
NewWitness
relies on retrieving the parent header from the chain. If the chain’s data is missing or corrupted, you return an error. Ensure you handle additional corner cases, such as the case when context.Number
is zero (genesis block).
90-99
: Add concurrency tests
AddState
uses a lock to combine MPT nodes. It would be beneficial to add targeted concurrency tests to confirm that different goroutines adding states simultaneously do not cause race conditions or inconsistencies.
core/state/database.go (1)
108-110
: LGTM: Clean interface extension
The new Witness() method is a well-defined addition to the Trie interface, providing a clear contract for implementations to expose accessed trie nodes.
trie/secure_trie.go (2)
193-193
: LGTM: Essential tracer state preservation
Correctly preserves the tracer state during trie copying, which is essential for maintaining witness tracking functionality.
226-229
: LGTM: Clean delegation pattern
The Witness implementation correctly delegates to the underlying trie's implementation, maintaining the abstraction hierarchy.
trie/zk_trie_test.go (1)
332-338
: LGTM: Clean storage equality check
The storage equality check is well-implemented and handles the RLP encoding correctly.
rollup/pipeline/pipeline.go (1)
231-231
: LGTM: StartPrefetcher parameter update
The addition of the nil
parameter aligns with the witness parameter addition to StartPrefetcher
. Since this is in a mining context, not collecting witness data is appropriate.
core/state/state_object.go (1)
503-514
: LGTM: Improved CodeSize implementation
The changes improve the implementation by:
- Adding proper database dependency injection
- Following a clear size calculation hierarchy
- Enhancing error handling with detailed messages
internal/web3ext/web3ext.go (2)
485-491
: LGTM: Added debug_executionWitness method
The new method properly extends the debug namespace with execution witness functionality. The use of null
input formatter is appropriate for raw block identifier input.
952-958
: LGTM: Added scroll_diskRoot method
The new method properly extends the scroll namespace with disk root retrieval functionality. The use of inputDefaultBlockNumberFormatter
is appropriate for block number parameter.
eth/api.go (1)
326-337
: LGTM: ExecutionWitness implementation is robust and well-structured.
The function properly handles error cases and provides clear error messages.
core/state/statedb.go (1)
110-112
: LGTM: Clean witness field addition.
The witness field is properly documented and integrated into the StateDB struct.
params/config.go (3)
40-41
: LGTM: Genesis state constants are properly defined.
The genesis state hashes are clearly defined as constants.
899-902
: LGTM: Euclid fork timing implementation follows established patterns.
The IsEuclid implementation matches other time-based fork checks.
669-671
: Consider adding validation for GenesisStateRoot.
The GenesisStateRoot field should be validated when used in chain initialization.
cmd/utils/flags.go (2)
186-189
: LGTM! The flag definition is clear and well-structured.
The new ScrollMPTFlag
follows the established pattern for boolean flags and provides clear usage documentation.
1886-1893
: LGTM! The Scroll Sepolia network configuration is correct.
The code properly configures Zktrie usage and enforces the required archive mode when Zktrie is enabled. Note that UseZktrie
is set to the inverse of ScrollMPTFlag
, which is logical as the flag controls MPT usage, not Zktrie usage.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
🧹 Nitpick comments (2)
rollup/ccc/async_checker.go (1)
101-105
: Add debug-level logging to clarify skipping of Euclid blocks.Currently, the CCC check is silently skipped for blocks identified as Euclid, which can obscure debugging if issues arise after this fork transition. Consider adding a log statement to more clearly indicate that the block is being skipped.
params/config.go (1)
899-903
: Fix doc comment to reference Euclid instead of Darwin.The doc block incorrectly refers to Darwin blocks, but the function checks EuclidTime. Update the comment for clarity and correctness.
-// IsEuclid returns whether num is either equal to the Darwin fork block or greater. +// IsEuclid returns whether the current timestamp is after the Euclid fork time.
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (3)
miner/scroll_worker.go
(3 hunks)params/config.go
(8 hunks)rollup/ccc/async_checker.go
(1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
- miner/scroll_worker.go
🔇 Additional comments (6)
params/config.go (6)
32-41
: No concerns about newly introduced genesis hashes.The definitions for additional genesis hashes and states appear correct. Ensure they are tracked accurately in any required documentation or test coverage.
345-345
: Valid pointer assignment for ScrollSepoliaChainConfig.This reference to ScrollSepoliaGenesisState looks correct. No further concerns.
386-386
: Valid pointer assignment for ScrollMainnetChainConfig.This reference to ScrollMainnetGenesisState looks correct. No further concerns.
669-671
: GenesisStateRoot field looks properly declared.This field is consistent with other Scroll config parameters. No immediate concerns about correctness or usage.
Line range hint
1116-1142
: New “IsEuclid” Rules field is properly added.The logic for setting and returning
IsEuclid
mirrors existing patterns for previous forks. Looks good.
640-640
: Verify alignment with other time-based forks.Introducing the EuclidTime field aligns well with other time-based forks, but ensure it’s consistently checked across the codebase (e.g., anywhere Darwin is used) to avoid potential misconfigurations.
✅ Verification successful
Let me verify the usage of fork times further by checking for any fork-specific logic implementations.
Fork time implementation is consistent and properly integrated
The verification shows that
EuclidTime
is well-aligned with other time-based forks in the codebase:
- The implementation follows the same pattern as other forks (Darwin, DarwinV2) with consistent helper methods (
IsEuclid
,IsDarwin
, etc.)- The fork time is properly integrated into the chain rules and used in critical components:
rollup/ccc/async_checker.go
: Handles Euclid-specific MPT logicminer/scroll_worker.go
: Manages state commitment based on Euclid activationcore/block_validator.go
: Validates state roots considering Euclid's requirementscore/vm/interpreter.go
: Fork-specific instruction sets are properly handledThe implementation maintains consistency with existing fork-time patterns and is correctly utilized across relevant components.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # This script searches for references to DarwinTime or EuclidTime throughout the repository # to verify consistent usage of fork time logic. rg -A 3 "DarwinTime|EuclidTime"Length of output: 3585
Script:
#!/bin/bash # Search for fork-specific rules or implementations ast-grep --pattern 'func $_($$) bool { return isForked$$$ }' # Also check for any direct comparisons or checks of these times rg "IsDarwin|IsEuclid" -A 3Length of output: 2351
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
🧹 Nitpick comments (2)
eth/api.go (2)
340-367
: Consider optimizing witness generation performanceThe
generateWitness
function includes proper error handling and validation. However, consider the following improvements:
- Add context parameter for cancellation support
- Consider caching the witness to avoid regeneration
- Document the special handling of
WithdrawTrieRootSlot
for maintainability-func generateWitness(blockchain *core.BlockChain, block *types.Block) (*stateless.Witness, error) { +func generateWitness(ctx context.Context, blockchain *core.BlockChain, block *types.Block) (*stateless.Witness, error) { witness, err := stateless.NewWitness(block.Header(), blockchain) if err != nil { return nil, fmt.Errorf("failed to create witness: %w", err) } + // Special handling: Collect storage for withdrawal processing statedb.GetState(rcfg.L2MessageQueueAddress, rcfg.WithdrawTrieRootSlot)
969-972
: Add documentation for DiskAndHeaderRoot structureConsider adding documentation explaining:
- The purpose of each field
- When they might differ
- Usage scenarios
+// DiskAndHeaderRoot contains both the disk state root and header state root. +// They may differ when the state is stored in a different format on disk +// compared to how it's represented in the block header. type DiskAndHeaderRoot struct { + // DiskRoot is the root hash of the state as stored on disk DiskRoot common.Hash `json:"diskRoot"` + // HeaderRoot is the root hash as stored in the block header HeaderRoot common.Hash `json:"headerRoot"` }
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (1)
eth/api.go
(3 hunks)
🔇 Additional comments (5)
eth/api.go (5)
37-37
: LGTM: Required imports added for new functionalityThe new imports for
stateless
andrcfg
packages are necessary for the implementation of execution witness and disk root functionality.Also applies to: 44-44
327-338
: LGTM: Well-structured API method with proper error handlingThe
ExecutionWitness
method properly handles block retrieval and error cases. The error messages are descriptive and helpful for debugging.
400-428
: LGTM: Well-designed witness data structuresThe
ExecutionWitness
structure and conversion utilities are well-designed:
- Clear JSON field tags
- Efficient map transformations
- Good documentation explaining future considerations
974-994
: LGTM: Well-implemented root retrievalThe
DiskRoot
method properly handles:
- Block retrieval errors
- Missing block cases
- Fallback to header root when disk root is not available
369-398
: Verify witness testing implementationThe
testWitness
function performs a complete reprocessing of the block to validate the witness. This could be computationally expensive for large blocks.Consider implementing a more efficient validation strategy for production environments, such as:
- Sampling-based validation
- Periodic validation instead of per-block
- Configurable validation depth
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
🧹 Nitpick comments (1)
core/state/snapshot/generate.go (1)
Line range hint
621-626
: Consider documenting the account serialization strategy.The exclusion of
PoseidonCodeHash
andCodeSize
from RLP encoding while still passing them toSlimAccountRLP
suggests a specific serialization strategy. Consider adding documentation to explain:
- Why these fields are excluded from RLP encoding
- How these fields are handled in the slim format
- The relationship between regular RLP and slim account formats
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (1)
core/state/snapshot/generate.go
(1 hunks)
🔇 Additional comments (2)
core/state/snapshot/generate.go (2)
621-622
: LGTM! Verify account serialization compatibility.The addition of
rlp:"-"
tags forPoseidonCodeHash
andCodeSize
fields is correct for excluding them from RLP encoding. This aligns with the L1 MPT support objective.Run this script to verify account serialization compatibility:
✅ Verification successful
Verified: RLP tags are correctly applied and consistent.
The
rlp:"-"
tags forPoseidonCodeHash
andCodeSize
fields in the snapshot account struct are consistent with the mainStateAccount
struct incore/types/state_account.go
. The RLP encoding/decoding operations across the codebase handle these fields appropriately.🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Check for any other account struct definitions and their RLP handling # to ensure consistency across the codebase # Search for other account struct definitions ast-grep --pattern 'type $_ struct { $$$ PoseidonCodeHash $_ $$$ }' # Search for account serialization/deserialization logic rg -A 5 "rlp\.(En|De)code.*account"Length of output: 10485
621-622
: Ensure snapshot generation tests cover new account format.Given these changes affect account serialization in the snapshot generation path, ensure that:
- Tests cover accounts with and without PoseidonCodeHash
- Snapshot generation and recovery scenarios are tested
- Integration tests verify L1 MPT compatibility
Run this script to check test coverage:
✅ Verification successful
Test coverage for new account format is comprehensive
The codebase includes extensive test coverage for the new account format including PoseidonCodeHash:
- Account serialization tests with both hash types
- Snapshot generation and recovery tests
- Edge cases and error handling tests
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Check for relevant test coverage # Look for snapshot generation tests rg -A 5 "Test.*Snapshot.*Generation" # Look for account serialization tests rg -A 5 "Test.*Account.*RLP"Length of output: 69
Script:
#!/bin/bash # Find and examine test files in core/state/snapshot fd -e go test snapshot core/state/snapshot # Look for snapshot-related tests with broader patterns rg -l "func Test.*Snapshot" core/state/snapshot # Search for account-related test code rg -A 5 "Account.*{" core/state/snapshotLength of output: 496
Script:
#!/bin/bash # Examine generate_test.go for relevant test cases cat core/state/snapshot/generate_test.go # Search for account-related tests with fixed pattern rg -A 5 "Account.*\{" core/state/snapshot/generate_test.go # Look for RLP-related test cases rg -A 5 "func.*Test.*RLP" core/state/snapshot/generate_test.goLength of output: 56418
b568657
to
cbe14fc
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 1
♻️ Duplicate comments (4)
core/stateless/witness.go (1)
116-122
:⚠️ Potential issueAdd nil check for Headers slice.
The
Root
method could panic ifHeaders
is empty.Apply this diff to add safety checks:
func (w *Witness) Root() common.Hash { + if len(w.Headers) == 0 { + return common.Hash{} + } return w.Headers[0].Root }trie/zk_trie_test.go (2)
274-277
:⚠️ Potential issueCritical: Remove hardcoded paths.
Using hardcoded filesystem paths will cause test failures in different environments.
Apply this diff to use temporary directories:
- zkDb, err := leveldb.New("/Users/omer/Documents/go-ethereum/l2geth-datadir/geth/chaindata", 0, 0, "", true) - mptDb, err := leveldb.New("/Users/omer/Documents/go-ethereum/l2geth-datadir-mpt/geth/chaindata", 0, 0, "", true) + tempDir, err := ioutil.TempDir("", "trie-test") + require.NoError(t, err) + defer os.RemoveAll(tempDir) + + zkDbPath := filepath.Join(tempDir, "zk") + mptDbPath := filepath.Join(tempDir, "mpt") + + zkDb, err := leveldb.New(zkDbPath, 0, 0, "", true) + require.NoError(t, err) + mptDb, err := leveldb.New(mptDbPath, 0, 0, "", true) + require.NoError(t, err)
294-294
: 🛠️ Refactor suggestionRemove global state from tests.
Using global variables in tests can cause race conditions in parallel tests and makes the tests harder to maintain.
Apply this diff to encapsulate the state:
+type testContext struct { + accountsLeft int + dbs *dbs +} -var accountsLeft = -1core/state/state_object.go (1)
547-549
:⚠️ Potential issueCritical: PoseidonCodeHash panic needs revision.
The panic in PoseidonCodeHash is problematic as this method is still being called by various parts of the codebase.
Consider returning an error instead of panicking:
func (s *stateObject) PoseidonCodeHash() []byte { if !s.db.IsZktrie() { - panic("PoseidonCodeHash is only available in zktrie mode") + return nil } return s.data.PoseidonCodeHash }🧰 Tools
🪛 GitHub Actions: CI
[error] 548-548: PoseidonCodeHash functionality not available outside of zktrie mode causing multiple test failures
🧹 Nitpick comments (10)
core/rawdb/accessors_state.go (2)
98-102
: Add function documentation.Please add a documentation comment explaining the purpose of this function and its parameters, following the Go documentation style used throughout the codebase.
Add the following documentation:
+// WriteDiskStateRoot stores the mapping between a header root and its corresponding disk state root. +// This mapping is crucial for L1 MPT support where the disk state might differ from the header state. func WriteDiskStateRoot(db ethdb.KeyValueWriter, headerRoot, diskRoot common.Hash) {
104-110
: Add function documentation and data validation.
- Please add documentation for this function.
- Consider validating the data length before conversion to prevent potential panics.
+// ReadDiskStateRoot retrieves the disk state root associated with the given header root. +// It returns an error if the mapping doesn't exist or if there's a database error. func ReadDiskStateRoot(db ethdb.KeyValueReader, headerRoot common.Hash) (common.Hash, error) { data, err := db.Get(diskStateRootKey(headerRoot)) if err != nil { return common.Hash{}, err } + if len(data) != common.HashLength { + return common.Hash{}, fmt.Errorf("invalid disk state root length: got %d, want %d", len(data), common.HashLength) + } return common.BytesToHash(data), nil }core/stateless/witness.go (1)
31-34
: Consider adding method documentation.The
HeaderReader
interface methods would benefit from documentation explaining the expected behavior, error conditions, and any constraints on the input parameters.trie/trie.go (2)
66-67
: Add documentation for tracer field.The new tracer field would benefit from documentation explaining its purpose and usage.
536-540
: Consider adding error handling for tracer.The onRead call should handle potential errors from the tracer.
Apply this diff:
rlp, err := t.db.Node(hash) if err != nil { return nil, err } - t.tracer.onRead(prefix, rlp) + if err := t.tracer.onRead(prefix, rlp); err != nil { + return nil, fmt.Errorf("tracer error: %w", err) + } return node, nileth/api.go (2)
340-367
: Consider adding retry mechanism for witness generation.The witness generation process could fail due to transient issues. Consider implementing a retry mechanism with backoff for improved reliability.
func generateWitness(blockchain *core.BlockChain, block *types.Block) (*stateless.Witness, error) { + maxRetries := 3 + var lastErr error + for i := 0; i < maxRetries; i++ { witness, err := stateless.NewWitness(block.Header(), blockchain) if err != nil { + lastErr = err + if i < maxRetries-1 { + time.Sleep(time.Duration(i+1) * 100 * time.Millisecond) + continue + } return nil, fmt.Errorf("failed to create witness: %w", err) } // ... rest of the function + } + return nil, fmt.Errorf("failed to create witness after %d retries: %w", maxRetries, lastErr) }
369-398
: Consider caching test results for performance optimization.The witness testing process could be optimized by caching test results for recently validated witnesses.
+var ( + witnessTestCache = make(map[common.Hash]error) + witnessTestCacheMu sync.RWMutex +) + func testWitness(blockchain *core.BlockChain, block *types.Block, witness *stateless.Witness) error { + witnessTestCacheMu.RLock() + if err, ok := witnessTestCache[block.Hash()]; ok { + witnessTestCacheMu.RUnlock() + return err + } + witnessTestCacheMu.RUnlock() + // ... existing implementation ... + + witnessTestCacheMu.Lock() + witnessTestCache[block.Hash()] = err + witnessTestCacheMu.Unlock() + return err }core/state/statedb.go (1)
300-302
: Consider consolidating code witness collection.The code witness collection is duplicated in GetCode and GetCodeSize. Consider extracting it to a helper method.
+func (s *StateDB) addCodeWitness(code []byte) { + if s.witness != nil { + s.witness.AddCode(code) + } +} + func (s *StateDB) GetCode(addr common.Address) []byte { stateObject := s.getStateObject(addr) if stateObject != nil { - if s.witness != nil { - s.witness.AddCode(stateObject.Code(s.db)) - } - return stateObject.Code(s.db) + code := stateObject.Code(s.db) + s.addCodeWitness(code) + return code } return nil }Also applies to: 311-313
cmd/utils/flags.go (2)
1886-1893
: LGTM! The configuration logic is robust and enforces necessary constraints.The code correctly:
- Configures the trie implementation based on the flag
- Enforces archive mode when using Zktrie
- Provides clear error messages and logging
Consider adding a comment explaining why archive mode is required for Zktrie.
cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) if cfg.Genesis.Config.Scroll.UseZktrie { - // disable pruning + // Zktrie requires archive mode as pruning is not supported if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { log.Crit("Must use --gcmode=archive") } log.Info("Pruning disabled") cfg.NoPruning = true }
1905-1912
: Consider refactoring to eliminate code duplication.The trie configuration logic is identical for both Scroll Sepolia and mainnet networks. Consider extracting this into a helper function to improve maintainability and reduce duplication.
+func configureScrollTrie(ctx *cli.Context, cfg *ethconfig.Config) { + cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) + if cfg.Genesis.Config.Scroll.UseZktrie { + // Zktrie requires archive mode as pruning is not supported + if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { + log.Crit("Must use --gcmode=archive") + } + log.Info("Pruning disabled") + cfg.NoPruning = true + } +} case ctx.GlobalBool(ScrollSepoliaFlag.Name): // ... other Sepolia config ... - cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) - if cfg.Genesis.Config.Scroll.UseZktrie { - if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { - log.Crit("Must use --gcmode=archive") - } - log.Info("Pruning disabled") - cfg.NoPruning = true - } + configureScrollTrie(ctx, cfg) case ctx.GlobalBool(ScrollFlag.Name): // ... other mainnet config ... - cfg.Genesis.Config.Scroll.UseZktrie = !ctx.GlobalBool(ScrollMPTFlag.Name) - if cfg.Genesis.Config.Scroll.UseZktrie { - if ctx.GlobalString(GCModeFlag.Name) != GCModeArchive { - log.Crit("Must use --gcmode=archive") - } - log.Info("Pruning disabled") - cfg.NoPruning = true - } + configureScrollTrie(ctx, cfg)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (30)
cmd/geth/main.go
(1 hunks)cmd/geth/usage.go
(1 hunks)cmd/utils/flags.go
(3 hunks)core/block_validator.go
(1 hunks)core/blockchain.go
(3 hunks)core/genesis.go
(1 hunks)core/rawdb/accessors_state.go
(1 hunks)core/rawdb/schema.go
(2 hunks)core/state/database.go
(2 hunks)core/state/snapshot/generate.go
(1 hunks)core/state/state_object.go
(2 hunks)core/state/state_test.go
(4 hunks)core/state/statedb.go
(8 hunks)core/stateless/database.go
(1 hunks)core/stateless/encoding.go
(1 hunks)core/stateless/witness.go
(1 hunks)core/types/state_account.go
(1 hunks)eth/api.go
(3 hunks)internal/web3ext/web3ext.go
(2 hunks)light/trie.go
(1 hunks)miner/scroll_worker.go
(6 hunks)params/config.go
(8 hunks)rollup/ccc/async_checker.go
(1 hunks)rollup/pipeline/pipeline.go
(1 hunks)trie/database.go
(2 hunks)trie/secure_trie.go
(2 hunks)trie/tracer.go
(1 hunks)trie/trie.go
(9 hunks)trie/zk_trie.go
(1 hunks)trie/zk_trie_test.go
(2 hunks)
🚧 Files skipped from review as they are similar to previous changes (19)
- cmd/geth/usage.go
- cmd/geth/main.go
- core/rawdb/schema.go
- core/genesis.go
- rollup/ccc/async_checker.go
- core/state/database.go
- core/types/state_account.go
- core/stateless/database.go
- core/block_validator.go
- core/state/state_test.go
- trie/zk_trie.go
- core/state/snapshot/generate.go
- rollup/pipeline/pipeline.go
- light/trie.go
- trie/database.go
- core/stateless/encoding.go
- miner/scroll_worker.go
- core/blockchain.go
- trie/tracer.go
🧰 Additional context used
🪛 GitHub Actions: CI
core/state/state_object.go
[error] 548-548: PoseidonCodeHash functionality not available outside of zktrie mode causing multiple test failures
🔇 Additional comments (23)
core/rawdb/accessors_state.go (3)
98-102
: Implementation looks good!The function follows the established patterns in the codebase for database writes and error handling.
104-110
: Implementation looks good!The function follows the established patterns in the codebase for database reads and error handling.
98-110
: Verify the integration with genesis block creation and key generation.Let's ensure the new functions are properly integrated with the genesis block creation and that the key generation function exists.
✅ Verification successful
Integration verified successfully
The functions are properly integrated across the codebase:
- Used correctly in genesis block creation
- Integrated with block processing and state management
- Key generation function exists and is correctly implemented
- Proper error handling and state consistency maintained
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Verify the integration of disk state root functions # Check the usage in genesis block creation echo "Checking usage in genesis.go..." rg "WriteDiskStateRoot|ReadDiskStateRoot" -A 5 -B 5 # Verify the existence of diskStateRootKey function echo "Checking diskStateRootKey definition..." ast-grep --pattern 'func diskStateRootKey(headerRoot common.Hash) []byte' # Check for any tests covering these functions echo "Checking for tests..." rg "TestWriteDiskStateRoot|TestReadDiskStateRoot"Length of output: 6057
core/stateless/witness.go (3)
82-88
: Consider using code hashes for storage efficiency.Storing raw bytecode in a map can be memory-intensive. Consider using code hashes as map keys instead.
90-99
: LGTM! Thread-safe state insertion.The implementation correctly uses mutex to ensure thread-safe state insertion.
71-80
:⚠️ Potential issueAdd bounds checking for block number.
The
AddBlockHash
method could panic ifnumber
is greater than the context number, as it would result in a negative slice index.Apply this diff to add safety checks:
func (w *Witness) AddBlockHash(number uint64) { + if w.context == nil { + return + } + if number > w.context.Number.Uint64() { + return + } for int(w.context.Number.Uint64()-number) > len(w.Headers) { tail := w.Headers[len(w.Headers)-1] w.Headers = append(w.Headers, w.chain.GetHeader(tail.ParentHash, tail.Number.Uint64()-1)) } }Likely invalid or redundant comment.
trie/secure_trie.go (2)
193-193
: LGTM! Proper deep copy of tracer.The Copy method correctly handles the tracer deep copy to prevent shared state between trie instances.
226-229
: LGTM! Clean witness implementation.The Witness method correctly delegates to the underlying trie's witness functionality.
core/state/state_object.go (1)
503-514
: LGTM! Improved code size retrieval.The CodeSize method now properly handles all cases:
- Returns cached code length if available
- Returns 0 for empty code hash
- Retrieves size from database otherwise
trie/trie.go (2)
320-324
: LGTM! Comprehensive node tracking.The implementation correctly tracks node operations:
- Insertion of new nodes
- Deletion of matched nodes
- Deletion during node merging
Also applies to: 396-400, 413-416, 478-482
621-631
: LGTM! Efficient witness implementation.The Witness method efficiently converts the access list to a map, with proper capacity pre-allocation.
internal/web3ext/web3ext.go (2)
485-490
: LGTM: New debug API method for execution witness retrieval.The
executionWitness
method is correctly integrated into the DebugJs module with proper parameter formatting.
952-957
: LGTM: New scroll API method for disk root retrieval.The
diskRoot
method is correctly integrated into the ScrollJs module with proper block number formatting.eth/api.go (2)
327-338
: LGTM: Well-structured execution witness retrieval with proper error handling.The implementation includes:
- Comprehensive error handling for block retrieval
- Clear error messages with context
- Efficient witness generation and transformation
969-994
: LGTM: Clean implementation of disk root retrieval.The implementation includes:
- Proper error handling for block retrieval
- Efficient disk root lookup with fallback
- Clear struct definition with JSON tags
core/state/statedb.go (4)
110-111
: LGTM: Clean addition of witness field.The witness field is properly documented and integrated into the StateDB struct.
166-174
: LGTM: Proper witness initialization in StartPrefetcher.The witness is correctly initialized and the existing prefetcher functionality is preserved.
934-947
: Potential redundant witness collection.The witness is collected twice for storage tries. Consider consolidating the witness collection to a single point.
992-996
: LGTM: Proper witness collection for account trie.The account trie witness is correctly collected after computing the hash.
params/config.go (3)
40-41
: LGTM: Well-defined genesis state roots.The genesis state roots are properly defined as constants for both Sepolia and Mainnet networks.
640-640
: LGTM: Clean implementation of Euclid upgrade.The Euclid upgrade is properly integrated with:
- Configuration field for timing
- Method to check activation status
- Consistent naming with other upgrades
Also applies to: 899-902
670-671
: LGTM: Clean addition of genesis state root support.The GenesisStateRoot field is properly added to ScrollConfig with appropriate JSON tags and documentation.
cmd/utils/flags.go (1)
186-189
: LGTM! The flag definition is clear and well-placed.The new boolean flag
ScrollMPTFlag
follows the existing pattern and provides a clear description of its purpose.
func checkTrieEquality(t *testing.T, dbs *dbs, zkRoot, mptRoot common.Hash, leafChecker func(*testing.T, *dbs, []byte, []byte)) { | ||
zkTrie, err := NewZkTrie(zkRoot, NewZktrieDatabase(dbs.zkDb)) | ||
require.NoError(t, err) | ||
|
||
mptTrie, err := NewSecure(mptRoot, NewDatabaseWithConfig(dbs.mptDb, &Config{Preimages: true})) | ||
require.NoError(t, err) | ||
|
||
expectedLeaves := zkTrie.CountLeaves() | ||
trieIt := NewIterator(mptTrie.NodeIterator(nil)) | ||
if accountsLeft == -1 { | ||
accountsLeft = int(expectedLeaves) | ||
} | ||
|
||
for trieIt.Next() { | ||
expectedLeaves-- | ||
preimageKey := mptTrie.GetKey(trieIt.Key) | ||
require.NotEmpty(t, preimageKey) | ||
leafChecker(t, dbs, zkTrie.Get(preimageKey), mptTrie.Get(preimageKey)) | ||
} | ||
require.Zero(t, expectedLeaves) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Add test cleanup.
The test should properly close and cleanup database resources.
Apply this diff to add cleanup:
func checkTrieEquality(t *testing.T, dbs *dbs, zkRoot, mptRoot common.Hash, leafChecker func(*testing.T, *dbs, []byte, []byte)) {
+ t.Cleanup(func() {
+ dbs.zkDb.Close()
+ dbs.mptDb.Close()
+ })
zkTrie, err := NewZkTrie(zkRoot, NewZktrieDatabase(dbs.zkDb))
require.NoError(t, err)
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.
func checkTrieEquality(t *testing.T, dbs *dbs, zkRoot, mptRoot common.Hash, leafChecker func(*testing.T, *dbs, []byte, []byte)) { | |
zkTrie, err := NewZkTrie(zkRoot, NewZktrieDatabase(dbs.zkDb)) | |
require.NoError(t, err) | |
mptTrie, err := NewSecure(mptRoot, NewDatabaseWithConfig(dbs.mptDb, &Config{Preimages: true})) | |
require.NoError(t, err) | |
expectedLeaves := zkTrie.CountLeaves() | |
trieIt := NewIterator(mptTrie.NodeIterator(nil)) | |
if accountsLeft == -1 { | |
accountsLeft = int(expectedLeaves) | |
} | |
for trieIt.Next() { | |
expectedLeaves-- | |
preimageKey := mptTrie.GetKey(trieIt.Key) | |
require.NotEmpty(t, preimageKey) | |
leafChecker(t, dbs, zkTrie.Get(preimageKey), mptTrie.Get(preimageKey)) | |
} | |
require.Zero(t, expectedLeaves) | |
} | |
func checkTrieEquality(t *testing.T, dbs *dbs, zkRoot, mptRoot common.Hash, leafChecker func(*testing.T, *dbs, []byte, []byte)) { | |
t.Cleanup(func() { | |
dbs.zkDb.Close() | |
dbs.mptDb.Close() | |
}) | |
zkTrie, err := NewZkTrie(zkRoot, NewZktrieDatabase(dbs.zkDb)) | |
require.NoError(t, err) | |
mptTrie, err := NewSecure(mptRoot, NewDatabaseWithConfig(dbs.mptDb, &Config{Preimages: true})) | |
require.NoError(t, err) | |
expectedLeaves := zkTrie.CountLeaves() | |
trieIt := NewIterator(mptTrie.NodeIterator(nil)) | |
if accountsLeft == -1 { | |
accountsLeft = int(expectedLeaves) | |
} | |
for trieIt.Next() { | |
expectedLeaves-- | |
preimageKey := mptTrie.GetKey(trieIt.Key) | |
require.NotEmpty(t, preimageKey) | |
leafChecker(t, dbs, zkTrie.Get(preimageKey), mptTrie.Get(preimageKey)) | |
} | |
require.Zero(t, expectedLeaves) | |
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
🧹 Nitpick comments (3)
core/blockchain_test.go (2)
3035-3037
: Handle errors to avoid potential test failures.While it's common in tests to ignore errors from
crypto.HexToECDSA
and other library calls, consider handling or at least logging them to simplify debugging if the key string becomes invalid in the future.
3043-3044
: Check the return value ofNewBlockChain
for errors.It's safer to handle or assert the returned error instead of discarding it (e.g.,
_ :=
). This practice helps avoid silently passing initialization issues, which can cause misleading test results.-blockchain, _ := NewBlockChain(db, nil, gspec.Config, engine, vm.Config{}, nil, nil) +blockchain, err := NewBlockChain(db, nil, gspec.Config, engine, vm.Config{}, nil, nil) +if err != nil { + t.Fatalf("failed to create blockchain: %v", err) +}miner/scroll_worker_test.go (1)
299-299
: Consider adding additional coverage for the non-zktrie path.All of these lines enable
chainConfig.Scroll.UseZktrie = true
across different test functions. If your intent is to maintain backward compatibility or support scenarios whereUseZktrie
might remain false, it may be worth introducing test cases that explicitly setUseZktrie = false
to ensure coverage of that path as well. This helps protect against regressions for users not enabling zktrie yet.Also applies to: 348-348, 409-409, 469-469, 532-532, 887-887, 1045-1045, 1090-1090, 1205-1205
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (5)
core/blockchain_test.go
(1 hunks)core/chain_makers.go
(2 hunks)core/genesis.go
(1 hunks)miner/scroll_worker_test.go
(9 hunks)trie/iterator_test.go
(1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
- core/genesis.go
⏰ Context from checks skipped due to timeout of 90000ms (1)
- GitHub Check: test
🔇 Additional comments (4)
trie/iterator_test.go (1)
299-299
: Use ofnewEmpty()
ensures proper Trie instantiation.
Replacing a zero-value declaration withnewEmpty()
clarifies the intentions of creating an initialized empty Trie instead of relying on a default struct value. This boosts readability and confidence that the trie is properly set up for testing.core/blockchain_test.go (1)
3042-3042
: Validate Zktrie usage in wider test coverage.Enabling
UseZktrie
is a key feature. Ensure that you have or plan to add test scenarios specifically covering Zktrie functionality (e.g., verifying root calculation, state transitions, etc.).core/chain_makers.go (2)
32-32
: No issues: The new import is relevant for MPT support
No immediate concerns. This import is consistent with the newly introduced L1 MPT approach, aligning with the pull request objective of supporting a Merkle Patricia Tree configuration.
268-268
: Ensure robust handling of config dependency
Ifconfig.Scroll
is ever nil orZktrieEnabled()
returns unexpected values, this could trigger runtime issues. Consider validating or documenting the usage contract to ensure safe toggling of Zktrie features.Here's a script to locate references to
ZktrieEnabled()
and confirm test coverage:
6540d08
to
4d0e694
Compare
4d0e694
to
a043d2f
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
♻️ Duplicate comments (5)
trie/zk_trie_test.go (3)
275-278
:⚠️ Potential issueCritical: Remove hardcoded filesystem paths.
The test uses hardcoded filesystem paths which will fail in different environments.
Apply this diff to fix test portability:
-zkDb, err := leveldb.New("/Users/omer/Documents/go-ethereum/l2geth-datadir/geth/chaindata", 0, 0, "", true) -mptDb, err := leveldb.New("/Users/omer/Documents/go-ethereum/l2geth-datadir-mpt/geth/chaindata", 0, 0, "", true) +tempDir, err := ioutil.TempDir("", "trie-test") +require.NoError(t, err) +defer os.RemoveAll(tempDir) + +zkDbPath := filepath.Join(tempDir, "zk") +mptDbPath := filepath.Join(tempDir, "mpt") + +zkDb, err := leveldb.New(zkDbPath, 0, 0, "", true) +require.NoError(t, err) +mptDb, err := leveldb.New(mptDbPath, 0, 0, "", true) +require.NoError(t, err)
295-295
:⚠️ Potential issueRemove global state from tests.
Using a global variable
accountsLeft
makes tests harder to maintain and can cause race conditions in parallel tests.Consider passing the state as a parameter or encapsulating it within a test context struct:
+type testContext struct { + accountsLeft int + dbs *dbs +} -var accountsLeft = -1
297-317
: 🛠️ Refactor suggestionAdd test cleanup.
The test should properly close and cleanup database resources.
Apply this diff to add cleanup:
func checkTrieEquality(t *testing.T, dbs *dbs, zkRoot, mptRoot common.Hash, leafChecker func(*testing.T, *dbs, []byte, []byte)) { + t.Cleanup(func() { + dbs.zkDb.Close() + dbs.mptDb.Close() + }) zkTrie, err := NewZkTrie(zkRoot, NewZktrieDatabase(dbs.zkDb)) require.NoError(t, err)core/state/statedb.go (1)
934-947
: 🛠️ Refactor suggestionRemove redundant witness collection.
As noted in the past review comments, the witness is being collected twice for storage tries. This should be consolidated to a single collection point.
- if s.witness != nil && obj.trie != nil { - s.witness.AddState(obj.trie.Witness()) - } obj.updateRoot(s.db) - if s.witness != nil && obj.trie != nil { - s.witness.AddState(obj.trie.Witness()) - } + if s.witness != nil && obj.trie != nil { + s.witness.AddState(obj.trie.Witness()) + }core/state/state_object.go (1)
547-549
:⚠️ Potential issueCritical: PoseidonCodeHash() panic needs reconsideration.
The panic in PoseidonCodeHash() when not in zktrie mode is problematic as this method is used by various parts of the codebase including state tracing, account proofs, snapshot generation, and state sync.
Consider one of these alternatives:
- Return the stored PoseidonCodeHash without the mode check
- Return a default value (like emptyPoseidonCodeHash) when not in zktrie mode
- Add a migration path for existing callers before enforcing this restriction
func (s *stateObject) PoseidonCodeHash() []byte { - if !s.db.IsZktrie() { - panic("PoseidonCodeHash is only available in zktrie mode") - } return s.data.PoseidonCodeHash }
🧹 Nitpick comments (6)
core/rawdb/accessors_state.go (2)
98-102
: Add documentation comments to explain the function's purpose.Following the package's documentation pattern, add a comment block explaining:
- The purpose of storing disk state roots
- The relationship between headerRoot and diskRoot
- When this function should be called
Add this documentation above the function:
+// WriteDiskStateRoot stores the mapping between a header's state root and its corresponding +// disk state root. This is used in L1 MPT support to track the relationship between +// the block header's state root and its actual disk representation. func WriteDiskStateRoot(db ethdb.KeyValueWriter, headerRoot, diskRoot common.Hash) {
104-110
: Consider aligning error handling with existing patterns.The error handling in
ReadDiskStateRoot
differs from other Read* functions in this file (e.g.,ReadPreimage
,ReadCode
) which ignore errors and return only the data. Consider:
- Either updating other Read* functions to include error handling (preferred for better error handling)
- Or aligning this function with the existing pattern for consistency
Option 1 (recommended) - Keep current implementation and update other functions:
-func ReadPreimage(db ethdb.KeyValueReader, hash common.Hash) []byte { - data, _ := db.Get(preimageKey(hash)) - return data +func ReadPreimage(db ethdb.KeyValueReader, hash common.Hash) ([]byte, error) { + return db.Get(preimageKey(hash)) }Option 2 - Align with existing pattern:
-func ReadDiskStateRoot(db ethdb.KeyValueReader, headerRoot common.Hash) (common.Hash, error) { - data, err := db.Get(diskStateRootKey(headerRoot)) - if err != nil { - return common.Hash{}, err - } - return common.BytesToHash(data), nil +func ReadDiskStateRoot(db ethdb.KeyValueReader, headerRoot common.Hash) common.Hash { + data, _ := db.Get(diskStateRootKey(headerRoot)) + return common.BytesToHash(data) }Also, add documentation:
+// ReadDiskStateRoot retrieves the disk state root corresponding to the given header root. +// This is used in L1 MPT support to look up the actual disk representation of a state root +// referenced in a block header. func ReadDiskStateRoot(db ethdb.KeyValueReader, headerRoot common.Hash) (common.Hash, error) {core/state/snapshot/account.go (1)
36-37
: Document the RLP field exclusion rationale.Consider adding a comment explaining why these fields are excluded from RLP encoding, as this helps future maintainers understand the design decision.
type Account struct { Nonce uint64 Balance *big.Int Root []byte KeccakCodeHash []byte + // PoseidonCodeHash and CodeSize are excluded from RLP encoding as they are derived fields + // used only for L1 MPT support and don't need to be persisted in snapshots PoseidonCodeHash []byte `rlp:"-"` CodeSize uint64 `rlp:"-"` }core/state/database.go (1)
142-144
: Consider caching the disk state root lookup.The disk state root lookup in
OpenTrie
could impact performance for repeated operations. Consider caching the result.func (db *cachingDB) OpenTrie(root common.Hash) (Trie, error) { + // Cache for disk state roots to avoid repeated lookups + if cachedRoot, ok := db.diskRootCache[root]; ok { + root = cachedRoot + } else if diskRoot, err := rawdb.ReadDiskStateRoot(db.db.DiskDB(), root); err == nil { + root = diskRoot + db.diskRootCache[root] = diskRoot + } - if diskRoot, err := rawdb.ReadDiskStateRoot(db.db.DiskDB(), root); err == nil { - root = diskRoot - }core/state/statedb.go (1)
300-302
: Consider consolidating duplicate witness collection logic.The witness collection logic is duplicated between
GetCode
andGetCodeSize
. Consider extracting this into a helper method to improve maintainability.+func (s *StateDB) addCodeToWitness(code []byte) { + if s.witness != nil { + s.witness.AddCode(code) + } +} func (s *StateDB) GetCode(addr common.Address) []byte { stateObject := s.getStateObject(addr) if stateObject != nil { - if s.witness != nil { - s.witness.AddCode(stateObject.Code(s.db)) - } + code := stateObject.Code(s.db) + s.addCodeToWitness(code) - return stateObject.Code(s.db) + return code } return nil } func (s *StateDB) GetCodeSize(addr common.Address) uint64 { stateObject := s.getStateObject(addr) if stateObject != nil { - if s.witness != nil { - s.witness.AddCode(stateObject.Code(s.db)) - } + code := stateObject.Code(s.db) + s.addCodeToWitness(code) return stateObject.CodeSize(s.db) } return 0 }Also applies to: 311-313
core/state/state_object.go (1)
503-514
: LGTM! Improved CodeSize implementation with proper error handling.The new implementation properly handles all cases with good optimization:
- Returns cached code size if available
- Quick returns 0 for empty code hash
- Falls back to database lookup with proper error handling
Consider making the error message slightly more specific:
- s.setError(fmt.Errorf("can't load code size %x: %v", s.KeccakCodeHash(), err)) + s.setError(fmt.Errorf("failed to load contract code size for hash %x: %v", s.KeccakCodeHash(), err))
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (41)
cmd/geth/main.go
(1 hunks)cmd/geth/usage.go
(1 hunks)cmd/utils/flags.go
(3 hunks)core/block_validator.go
(1 hunks)core/blockchain.go
(3 hunks)core/blockchain_test.go
(5 hunks)core/chain_makers.go
(3 hunks)core/genesis.go
(1 hunks)core/genesis_test.go
(1 hunks)core/rawdb/accessors_state.go
(1 hunks)core/rawdb/schema.go
(2 hunks)core/state/database.go
(2 hunks)core/state/snapshot/account.go
(1 hunks)core/state/snapshot/generate.go
(1 hunks)core/state/state_object.go
(2 hunks)core/state/state_test.go
(4 hunks)core/state/statedb.go
(8 hunks)core/stateless/database.go
(1 hunks)core/stateless/encoding.go
(1 hunks)core/stateless/witness.go
(1 hunks)core/types/state_account.go
(1 hunks)eth/api.go
(3 hunks)internal/web3ext/web3ext.go
(2 hunks)light/trie.go
(1 hunks)miner/scroll_worker.go
(6 hunks)miner/scroll_worker_test.go
(10 hunks)params/config.go
(10 hunks)rollup/ccc/async_checker.go
(1 hunks)rollup/ccc/async_checker_test.go
(1 hunks)rollup/pipeline/pipeline.go
(1 hunks)trie/database.go
(2 hunks)trie/iterator_test.go
(2 hunks)trie/proof.go
(1 hunks)trie/proof_test.go
(9 hunks)trie/secure_trie.go
(2 hunks)trie/secure_trie_test.go
(1 hunks)trie/tracer.go
(1 hunks)trie/trie.go
(9 hunks)trie/trie_test.go
(4 hunks)trie/zk_trie.go
(1 hunks)trie/zk_trie_test.go
(2 hunks)
🚧 Files skipped from review as they are similar to previous changes (25)
- cmd/geth/usage.go
- core/chain_makers.go
- light/trie.go
- cmd/geth/main.go
- core/genesis.go
- miner/scroll_worker.go
- trie/iterator_test.go
- rollup/ccc/async_checker.go
- core/types/state_account.go
- core/block_validator.go
- core/rawdb/schema.go
- trie/zk_trie.go
- trie/trie.go
- core/state/snapshot/generate.go
- core/stateless/database.go
- rollup/pipeline/pipeline.go
- trie/proof_test.go
- trie/database.go
- params/config.go
- trie/tracer.go
- core/stateless/witness.go
- eth/api.go
- core/blockchain.go
- core/stateless/encoding.go
- core/blockchain_test.go
⏰ Context from checks skipped due to timeout of 90000ms (1)
- GitHub Check: test
🔇 Additional comments (27)
internal/web3ext/web3ext.go (2)
485-490
: LGTM! The executionWitness method is well-integrated.The method follows the module's conventions and properly supports debugging capabilities for execution witnesses.
952-957
: LGTM! The diskRoot method is well-integrated.The method follows the module's conventions and properly supports disk-based MPT root retrieval with appropriate block number formatting.
trie/trie_test.go (4)
596-597
: Hash value update reflects L1 MPT changes.The updated hash value aligns with the new Layer 1 MPT implementation changes.
600-601
: Sequential hash updates maintain consistency.The sequence of hash value updates correctly reflects the state transitions in the new Layer 1 MPT implementation.
Also applies to: 604-605
628-628
: Updated commit hash aligns with L1 MPT.The new expected hash value correctly represents the trie's state after commit in the Layer 1 MPT implementation.
728-733
: Updated sequence hashes for varying trie sizes.The hash value updates correctly reflect the write and callback sequences for different trie sizes in the Layer 1 MPT implementation. The test coverage remains comprehensive, testing small (20), medium (200), and large (2000) element counts.
core/state/snapshot/account.go (1)
36-37
: Verify the implications of excluding fields from RLP encoding.The addition of
rlp:"-"
tags will excludePoseidonCodeHash
andCodeSize
from RLP encoding/decoding. While this appears intentional based on the PR objectives for L1 MPT support, we should verify:
- That these fields don't need to be persisted in snapshots
- That this change maintains backward compatibility with existing snapshots
✅ Verification successful
The RLP exclusion of these fields is intentional and correct.
The
PoseidonCodeHash
andCodeSize
fields are Scroll-specific extensions to the account structure that are intentionally excluded from RLP encoding as they are computed values that can be regenerated. The codebase properly handles these fields through:
- Consistent RLP exclusion pattern across related structs
- Proper initialization of empty states
- Dedicated conversion functions in the snapshot system
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Search for related structs to verify consistent RLP handling echo "Checking related account structs and their RLP handling..." rg -A 10 "type.*Account.*struct" --type go # Look for existing RLP usage patterns echo "Checking existing RLP encoding/decoding patterns..." rg "rlp\.(\w+)" --type goLength of output: 66676
Script:
#!/bin/bash # Check for direct field references and usage patterns echo "Checking field references..." rg "PoseidonCodeHash|CodeSize" --type go -B 2 -A 2 # Look for snapshot conversion code echo "Checking snapshot conversion code..." rg "SlimAccount|FullAccount" --type go -B 2 -A 2 # Check for Account type conversions echo "Checking Account conversions..." rg "Account.*convert|convert.*Account" --type go -B 2 -A 2Length of output: 66082
rollup/ccc/async_checker_test.go (2)
28-29
: LGTM! Good practice cloning the test config.Creating a local copy of the test chain config prevents side effects on the shared test configuration.
31-31
: Verify test coverage with ZK-trie enabled.The test now uses ZK-trie configuration. Please ensure that:
- The test covers all relevant ZK-trie specific behaviors.
- Edge cases specific to ZK-trie are tested.
Also applies to: 35-35, 39-41
trie/secure_trie_test.go (1)
115-115
: LGTM! Improved thread safety using deep copy.Replacing shallow copy with
Copy()
method prevents potential data races by ensuring each goroutine operates on an independent instance ofSecureTrie
.core/genesis_test.go (1)
44-44
: Verify the new genesis hash calculation.The genesis hash has been updated, likely due to L1 MPT changes. Please ensure:
- The new hash is correctly calculated with L1 MPT enabled.
- The hash calculation is deterministic across different environments.
✅ Verification successful
Test-specific genesis hash verified as deterministic ✓
The hash is used in a test case that explicitly verifies hash consistency across different initialization methods and database states. The test framework ensures deterministic hash calculation through multiple verification paths.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Search for genesis hash usage and verification rg -A 5 "0x89c99d90b79719238d2645c7642f2c9295246e80775b38cfd162b696817fbd50"Length of output: 506
Script:
#!/bin/bash # Search for genesis hash calculation and L1 MPT related code echo "=== Genesis Hash Calculation ===" ast-grep --pattern 'func (g *Genesis) Hash() common.Hash' echo -e "\n=== L1 MPT Related Code ===" rg -l "L1MPT|l1mpt" --type go echo -e "\n=== Genesis Test Setup ===" cat core/genesis_test.goLength of output: 8011
trie/proof.go (1)
562-562
: LGTM! Enhanced trie operation tracking.Adding tracer initialization improves observability of trie operations during range proof verification.
✅ Verification successful
Verified: Tracer initialization is consistent across all trie creation paths
The tracer is properly initialized in all cases:
- Direct Trie initialization in proof verification
- Base New() function implementation
- Derived creation methods (NewSecure)
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Verify consistent tracer initialization across trie operations ast-grep --pattern $'tr := &Trie{ $$$ tracer: $_ $$$ }'Length of output: 62
Script:
#!/bin/bash # Search for Trie struct initializations ast-grep --pattern '&Trie{$$$}' # Look for tracer-related definitions and usage rg "type.*[Tt]racer" -A 5 rg "func.*[Tt]racer" -A 5 # Find files containing tracer-related code rg "tracer" --type goLength of output: 66384
Script:
#!/bin/bash # Find all Trie struct initializations rg "&Trie{" --type go -A 3 # Find all newTrie function calls rg "newTrie" --type go -A 3Length of output: 2192
Script:
#!/bin/bash # Find Trie struct initializations with proper escaping rg "(?s)&Trie\{.*?\}" --type go -A 3 # Find NewTrie/NewSecure implementations rg "func (New|new)(Trie|Secure)" --type go -A 5Length of output: 1453
Script:
#!/bin/bash # Find New function implementation in trie package rg "^func New\(" --type go -A 10 trie/Length of output: 481
core/state/database.go (1)
108-110
: LGTM: Well-designed interface extension.The new
Witness
method is a clean addition to theTrie
interface that enables tracking of accessed trie nodes, which is valuable for debugging and optimization.trie/secure_trie.go (2)
193-193
: LGTM: Proper state handling in Copy.The addition of
tracer
copying ensures that the copied trie maintains its own state tracking.
226-229
: LGTM: Clean Witness implementation.The
Witness
implementation correctly delegates to the underlying trie's witness tracking.core/state/statedb.go (5)
110-112
: LGTM: New witness field added to StateDB.The addition of the
witness
field to store state witnesses for cross validation is well-structured and properly documented.
166-174
: LGTM: StartPrefetcher updated to support witness collection.The StartPrefetcher method is cleanly updated to handle witness initialization.
950-959
: LGTM: Witness collection for read-only operations.The witness collection for read-only operations on the account trie is properly implemented.
992-996
: LGTM: Witness collection for account trie.The witness collection for the account trie after computing the intermediate root is correctly implemented.
742-744
: LGTM: Proper witness handling in Copy method.The Copy method correctly handles the witness field by creating a deep copy when needed.
miner/scroll_worker_test.go (2)
227-235
: LGTM: Proper chain config cloning in test setup.The test setup now correctly clones the chain configuration to prevent modifications to the original config.
289-300
: LGTM: Consistent chain config setup across tests.The chain configuration setup is consistent with other tests, properly cloning configs and setting required parameters.
cmd/utils/flags.go (2)
186-189
: LGTM: Well-defined MPT flag.The ScrollMPTFlag is properly defined with clear usage description.
1886-1893
: LGTM: Proper enforcement of archive mode for Zktrie.The code correctly enforces archive mode when using Zktrie and provides clear user feedback.
The implementation:
- Properly sets UseZktrie based on the ScrollMPTFlag
- Enforces archive mode when using Zktrie
- Provides clear error messages
- Logs important configuration changes
Also applies to: 1905-1912
core/state/state_test.go (3)
158-159
: LGTM! Improved initialization clarity.The change makes the database initialization more explicit by separating it from state creation, which is a good practice for better code readability and dependency management.
205-205
: LGTM! Correctly updated function call.The change properly passes the database parameter to align with the updated compareStateObjects function signature.
214-214
: LGTM! Function signature and usage updated consistently.The compareStateObjects function has been properly updated to handle the new CodeSize method signature, maintaining consistency with the changes in state_object.go.
Also applies to: 233-234
1. Purpose or design rationale of this PR
best reviewed commit by commit.
2. PR title
Your PR title must follow conventional commits (as we are doing squash merge for each PR), so it must start with one of the following types:
3. Deployment tag versioning
Has the version in
params/version.go
been updated?4. Breaking change label
Does this PR have the
breaking-change
label?Summary by CodeRabbit
Based on the comprehensive summary, here are the updated release notes:
New Features
Performance Improvements
Configuration Updates
Debugging Capabilities
Blockchain Compatibility
These release notes capture the high-level changes without revealing implementation details, focusing on user-facing improvements and new capabilities.